- 
                Notifications
    You must be signed in to change notification settings 
- Fork 346
adds search icon to inbox and combined feed pages #1864
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: main
Are you sure you want to change the base?
Conversation
| Need your review on this. Do let me know if tests are also required for this. | 
| Thanks! Yeah, let's have a few quick test cases for this: just checking that 
 | 
| Sure, will add the tests and update the PR. | 
| Thanks for the detailed screenshots! The placement looks good to me. | 
| Hey, @gnprice , @chrisbobbe . I have added the following tests for the search icon button in this PR : Positive Cases - Search Button Should Appear 
 Negative Cases - Search Button Should NOT Appear 
 | 
| Hey Guys, just a sincere remainder. This PR needs a review. Thanks. cc : @gnprice , @chrisbobbe | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks, this looks helpful! Comments below, and please organize the commits (and write their commit messages) according to our project style.
        
          
                lib/widgets/message_list.dart
              
                Outdated
          
        
      | actions.add(IconButton( | ||
| icon: const Icon(ZulipIcons.search), | ||
| onPressed: () => Navigator.push(context, | ||
| MessageListPage.buildRoute(context: context, | ||
| narrow: KeywordSearchNarrow(''))), | ||
| )); | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could you add a tooltip:  here, like on the "Channel feed" button for topic narrows?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also, let's add this in a separate commit from the commit that changes the inbox page. Please ask in #git help in the development community if you need help doing that.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Added the tooltip: zulipLocalizations.searchMessagesPageTitle, because this same tooltip was being used for the search button (_SearchButton) inside the main menu.
        
          
                lib/widgets/message_list.dart
              
                Outdated
          
        
      | narrow: KeywordSearchNarrow(''))), | ||
| )); | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also nit:
| narrow: KeywordSearchNarrow(''))), | |
| )); | |
| narrow: KeywordSearchNarrow(''))))); | 
(like in similar code)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done
        
          
                lib/widgets/home.dart
              
                Outdated
          
        
      | actions: _tab.value == _HomePageTab.inbox ? [ | ||
| IconButton( | ||
| icon: const Icon(ZulipIcons.search), | ||
| onPressed: () => Navigator.of(context).push(MessageListPage.buildRoute( | ||
| context: context, narrow: KeywordSearchNarrow(''))), | ||
| ), | ||
| ] : null), | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For the value we pass as actions, I think it would be neater as _currentTabAppBarActions, on the pattern of _currentTabTitle.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Added _currentTabAppBarActions, following the pattern of _currentTabTitle
        
          
                test/widgets/home_test.dart
              
                Outdated
          
        
      | // to an ancient server: | ||
| // https://github.com/zulip/zulip-flutter/pull/1410#discussion_r1999991512 | ||
|  | ||
| group('search button', () { | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think a neater way to cover the inbox page in tests would be:
- Don't add this new 'search button' group. (The search button isn't really a feature of the whole home page, just the inbox.)
- To test that the button is present on the "Inbox" tab and not the other tabs, update and rename the existing test 'update app bar title when switching between views' in the 'bottom nav navigation' group.
- In test/widgets/inbox_test.dart, add a test that the button does the correct thing when tapped.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Made the changes you suggested.
Thanks for pointing out, made me go through tests again which helped me understand the structures of these tests even more.
        
          
                test/widgets/message_list_test.dart
              
                Outdated
          
        
      | group('app bar search button', () { | ||
| // Tests for the search icon button that should appear in the app bar | ||
| // on the combined feed screen but not on other message list screens. | ||
| // We check the app bar's actions list directly to avoid confusion with | ||
| // the search icon that appears in the search input field. | ||
|  | ||
| testWidgets('search icon button appears on CombinedFeedNarrow', (tester) async { | ||
| // Set up the combined feed (all messages) view | ||
| await setupMessageListPage(tester, narrow: const CombinedFeedNarrow(), messages: []); | ||
|  | ||
| // Verify that the search button is present in the app bar actions | ||
| final appBar = tester.widget<ZulipAppBar>(find.byType(ZulipAppBar)); | ||
| final actions = appBar.actions ?? []; | ||
| final hasSearchButton = actions.any((widget) => | ||
| widget is IconButton && | ||
| widget.icon is Icon && | ||
| (widget.icon as Icon).icon == ZulipIcons.search); | ||
| check(hasSearchButton).isTrue(); | ||
| }); | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think we don't need tests to check that the button is absent for various narrows. I think there's almost no risk that we'll accidentally create a bug that such tests would catch, which means the tests aren't very useful. Here's a helpful talk by Greg covering that kind of advice and more: #learning > Greg's talk at FlutterCon USA 2025 @ 💬
I think we just need one test that looks for the search button in "Combined feed" and checks that it does the correct thing when tapped. That test should go in the existing "app bar" group.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think there's a plausible bug where we could have had this button on all message-list pages (or all other than search narrows). So that's why in that linked comment I asked for one such test.
It looks like the revision this comment was on had many such tests, for different types of narrow. I agree that that's unnecessary, and it makes the overall set of tests harder to read.
3453701    to
    3608a3e      
    Compare
  
    | @chrisbobbe , just a heads up! Thank you so much for the valuable feedback!! | 
| Hey @gnprice , @chrisbobbe | 
| Hey, @chrisbobbe, just a follow up on this Thanks!! | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks! Comments below, all small. I've left detailed feedback on the first commit message; please apply it to the second commit message too (and remember to say Fixes #1854. there too.)
        
          
                test/widgets/home_test.dart
              
                Outdated
          
        
      | // Helper to check if search button is present | ||
| bool hasSearchButton() { | ||
| final appBar = tester.widget<ZulipAppBar>(find.byType(ZulipAppBar)); | ||
| final actions = appBar.actions ?? []; | ||
| return actions.any((widget) => | ||
| widget is IconButton && | ||
| widget.icon is Icon && | ||
| (widget.icon as Icon).icon == ZulipIcons.search); | ||
| } | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The comment doesn't add new information; let's not add it.
Also this test looks more brittle than it needs to be, by depending on details of the implementation. For example, in the future, I could imagine replacing IconButton with ZulipIconButton in the app code. That would break some of these tests even though it would be a valid change.
How about a helper like this instead:
      final findSearchButton = find.descendant(
        of: find.byType(ZulipAppBar),
        matching: find.byIcon(ZulipIcons.search));then callers would do
check(findSearchButton).findsOne();or
check(findSearchButton).findsNothing();. That would work equally well with either implementation.
This is an example of the principle "test at stable interfaces", in particular the user interface; see Greg's talk here for details: #learning > Greg's talk at FlutterCon USA 2025 @ 💬
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done
        
          
                test/widgets/home_test.dart
              
                Outdated
          
        
      | (widget.icon as Icon).icon == ZulipIcons.search); | ||
| } | ||
|  | ||
| // Inbox tab: should have title "Inbox" and search button | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
These comments don't add new information that's not in the code; let's not add them.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
removed
        
          
                test/widgets/inbox_test.dart
              
                Outdated
          
        
      | }); | ||
|  | ||
| testWidgets('tapping search button navigates to search page', (tester) async { | ||
| // Set up navigation tracking | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
These comments don't add new information that's not in the code; let's not add them.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
removed
| } | ||
| } | ||
|  | ||
| List<Widget>? get _currentTabAppBarActions { | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
home: Add search button and related tests.
Adds a search icon and tooltip to the icon in the home page app bar
that appears on the inbox tab. Uses the same localized 'Search' text
for consistency with the message list search button.
Also refactors the app bar actions logic by extracting it into a
_currentTabAppBarActions getter.
Updates the existing bottom nav navigation test inside home_test to verify search button presence
across different tabs: present on inbox, absent on channels and
direct messages tabs.
Also updates the existing InboxPage tests to verify the search button navigates to search page when tapped
Let's shorten the commit message down to just what's needed:
- Tests are always part of the process for building new features, so we don't normally mention them in commit messages.
- This one should say Fixes-partly: #1854, and the next one (which completes the fix) should sayFixes #1854.
- Let's mention the "Inbox" page in the summary line. There's room to do that, and it could be helpful for someone who's only reading the summary lines. Then the first sentence in the body is redundant and can be removed.
- The sentence about _currentTabAppBarActionsisn't quite accurate: it's not that we're really changing/refactoring existing logic; rather it's more like we're adding logic that wasn't there before. There isn't anything very remarkable about how we're doing this, so I would just remove the sentence.
- This sentence doesn't make sense to me: "Uses the same localized 'Search' text
 for consistency with the message list search button." At this commit, "message list search button" doesn't refer to anything that exists, so it's confusing. You could rewrite it to make the context clear, by mentioning the plan to add a similar button the "Combined feed", which is done in the next commit…but you don't need to; the reader can easily look at the code if they're worried abou an inconsistency.
So, putting all that together, it would be something like:
home: Add search button to Inbox page
Fixes-partly: #1854
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Updated the commit message as per suggestion.
        
          
                test/widgets/message_list_test.dart
              
                Outdated
          
        
      | }); | ||
|  | ||
| testWidgets('search button on combined feed navigates to search page', (tester) async { | ||
| // Set up navigation tracking | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
(See my earlier feedback about unnecessary code comments)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sure
3608a3e    to
    4772ff8      
    Compare
  
    | @chrisbobbe , Thanks! for the valuable feedback and review. | 
| Thanks, LGTM! Marking for Greg's review. | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks @komal-uniqode for adding this, and @chrisbobbe for the careful reviews! Just small comments below.
| appBar: ZulipAppBar(titleSpacing: 16, | ||
| title: Text(_currentTabTitle)), | ||
| title: Text(_currentTabTitle), | ||
| actions: _currentTabAppBarActions | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit:
| actions: _currentTabAppBarActions | |
| actions: _currentTabAppBarActions, | 
(or else put the closing ) on the same line)
| onPressed: () => Navigator.of(context).push(MessageListPage.buildRoute( | ||
| context: context, narrow: KeywordSearchNarrow(''))), | ||
| ), | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit:
| onPressed: () => Navigator.of(context).push(MessageListPage.buildRoute( | |
| context: context, narrow: KeywordSearchNarrow(''))), | |
| ), | |
| onPressed: () => Navigator.of(context).push(MessageListPage.buildRoute( | |
| context: context, narrow: KeywordSearchNarrow('')))), | 
| onPressed: () => Navigator.of(context).push(MessageListPage.buildRoute( | ||
| context: context, narrow: KeywordSearchNarrow(''))), | ||
| ), | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: more generally, make this look as similar as possible to the other navigation callback found in this same widget. That way the only substantive difference between them — the difference in what narrow to go to — stands out.
Similarly, make this look identical to the navigation callback added in your other commit, which is intended to have an identical effect.
In particular that means use Navigator.push(context, …) rather than Navigator.of(context).push(…). (They mean the same thing, so avoid making it look like there's a difference.) Also make the formatting look the same.
| check(find.text('DMs with Muted user, User 2, Muted user')).findsOne(); | ||
| }); | ||
|  | ||
| testWidgets('search button on combined feed navigates to search page', (tester) async { | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As I requested above (#1864 (comment)), let's also have a test that looks at a message list on some other narrow and checks that there isn't a search button there.
(But, as discussed in that subthread, not the whole suite of all the other narrow types.)
| List<Widget> actions = []; | ||
| switch (narrow) { | ||
| case CombinedFeedNarrow(): | ||
| actions.add(IconButton( | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit in commit message:
message_list: Add search button to combined feed
In our commit-message summary lines, the message list is so frequent that we use an abbreviation for it: msglist:.
| check(searchButtonFinder).findsOne(); | ||
|  | ||
| pushedRoutes.clear(); | ||
|  | ||
| connection.prepare(json: eg.newestGetMessagesResult( | ||
| foundOldest: true, messages: []).toJson()); | ||
|  | ||
| await tester.tap(searchButtonFinder); | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This can be simplified a bit: the check(searchButtonFinder).findsOne() is redundant with the fact that tester.tap on the same finder succeeds immediately after it.
| matching: find.byIcon(ZulipIcons.search)); | ||
| check(searchButtonFinder).findsOne(); | ||
|  | ||
| pushedRoutes.clear(); | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: move this to right after the step that pushed these routes (and then join it in the same stanza, with no separating blank line)
| check(pushedRoutes).single.isA<WidgetRoute>().page | ||
| .isA<MessageListPage>() | ||
| .initNarrow.equals(KeywordSearchNarrow('')); | ||
| await tester.pump(Duration.zero); | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Interesting — what's this final pump about? What happens if it's removed?
Added a search icon on inbox and the combined feed screen which opens up the usual search page.
Fixes : #1854
Screen.Recording.2025-09-20.at.2.11.43.PM.mov
Screen.Recording.2025-09-20.at.2.17.36.PM.mov